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Lua очень простой 


local function hello(name) 
print('Hello, ' .. пате .. '!') 
епд 


local names = {'World', 'Highload', "Мозсои" | 


for 1, пате in pairs(name) до 
hello(name) 
end 


Hello, World! 
Hello, Moscow! 
Hello, Highload! 


1^ HighLoad 
(NC оке 


Типизациа 


о Динамическаа 


o Не сильная (но почти) 


1A HighLoad++ 
(HL) de der 


Пример из жизни 


Гог 4410, new leader in pairs(new leaders) do 


log.info('Replicaset 95: new leader %s, was 95", 
uuid, 


describe(new leader), 
describe(old leaders[uuid]) 


1^ HighLoad 
(HD менен 


Пример из жизни 


Гог 4410, new leader in pairs(new leaders) до 


if uuid == my uuid then 
uuid = uuid .. ' (me)' 
end 


log.info('Replicaset 95: new leader 95, was 95", 
uuid, 
describe(new leader), 
describe(old leaders[uuid]) 


1^ HighLoad 
(нь) нген» 


Пример из жизни 


Гог 4410, new leader in pairs(new leaders) до 


if uuid == my uuid then 
uuid = uuid .. ' (me)' 
end 


log.info('Replicaset 95: new leader 95, was 95", 
uuid, 
describe(new leader), 


1^ HighLoad 
So 


Пример из жизни 


Гог 4410, new leader in pairs(new leaders) до 


+ local replicaset name = uuid 


if uuid == my uuid then 
uuid = uuid .. ' (me)' 
replicaset name - replicaset name .. ' (me)' 


end 


log.info('Replicaset 95: new leader 95, was 95", 
uuid, 
replicaset name, 
describe(new leader), 
describe(old leaders | 0141) 
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Типизациа — не сильная 


Нельзя 


"К" .. nil -- attempt to concatenate а nil value 
11, 2} + {3} -- attempt to perform arithmetic on a table value 


- Python: [1, 2] + [3] == [1, 2, 3] 
- JS: (1, 2] + [3] == |1,23! 
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Типизациа — не сильная 


Нельзя 


"К" .. nil -- attempt to concatenate a nil value 
(1, 2) + {3} -- attempt to perform arithmetic on a table value 


-- Python: [1, 2] + [3] == [1, 2, 3] 
-- JS: [1, 2] + [3] == '1,23' 


Можно 


math.sqrt("144") -- 12 (number) 


string. len(1337) -- 4 


Lua 51 Reference Manual. 82.2.1 Coercion 


1^ HighLoad 
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Аргументы надо проверять 


function get_stat(uri, opts) 


return http.get('http://' .. uri .. '/stat', opts) 


end 


1A HighLoad« 
(HL) tees 


Аргументы надо провератБ 


function get_stat(uri, opts) 
return http.get('http://' .. uri .. '/stat', opts) 


end 


get stat(req.uri) -- req.uri -- nil 
- error: api.lua:310: attempt to concatenate a nil value 
- He понатно 


1^ HighLoad 
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Аргументы надо проверять 


function get_stat(uri, opts) 
assert(type(uri) == 'string', 'uri must be a string’) 


end 


1A HighLoad+ 
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Аргументы надо проверять 


function get_stat(uri, opts) 
assert(type(uri) == 'string', 'uri must be a string’) 


end 


get stat(req.uri) -- req.uri == nil 
- error: api.lua:310: uri must Бе a string 
- Уже лучше 


1^ HighLoad 
(MD анн 


Аргументы надо провератБ 


function get_stat(uri, opts) 
assert(type(uri) == 'string', “игі must be a string’) 


end 


get stat(req.uri) -- req.uri == nil 
-- error: api.lua:310: uri must be a string 
-- Уже лучше 


get stat('localhost', {timeuot = 1}) 
ма ^^ typo 
-- Ошибки нет, но поведение не правильное 
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Аргументы надо проверять 


require('checks' ) 


function get_stat(uri, opts) 


checks('string', {timeout = '?number'}) 


end 
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(іш) анн 


Аргументы надо проверять 


require('checks' ) 


function get_stat(uri, opts) 


checks('string', {timeout = '?number']) 


end 


get_stat() 
-- error: bad argument #1 to get_stat (string expected, got nil) 
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Аргументы надо провератБ 


require('checks') 


function get stat(uri, opts) 


checks('string', [timeout = '?number']) 


end 


get stat() 
-- error: bad argument #1 to get stat (string expected, got nil) 


get stat('localhost', {timeuot = 1)) 


-- error: unexpected argument opts.timeuot to get stat 


1^ HighLoad 
oa 


Типизация — простая, как топор 
о роојеап, string, number 

о function, thread 

o Userdata, 

о table 

о nil 
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number == double 


1A HighLoad++ 
(HL) de der 


number == double 


assert equals(0.1 + 0.2, 0.3) 


- error: 
- expected: 0.3 
actual: 0.3 
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number == double 


assert_equals(0.1 + 0.2, 0.3) 
-- error: 

-- expected: 0.3 

-- actual: 0.3 


2753 - 2 -- 9007199254740990 
2753 - 1 -- 9007199254740991 


2753 + 2 -- 9007199254740994 


2753 + 1 == 2753 -- true 
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number == double 


assert_equals(0.1 + 0.2, 0.3) 
-- error: 

-- expected: 0.3 

-- actual: 0.3 


2753 - 2 -- 9007199254740990 
2753 - 1 -- 9007199254740991 


2753 + 2 -- 9007199254740994 


2753 + 1 == 2753 -- true 


now = clock.time64() -- 1621068872741010434, ~2760 
ffi.typeof(t) -- ctype<uint64 t» 
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ТЕЕЕ 754 


ехропеп! fraction 
sign (11 bit) (52 bit) 


I 1-1 
шшшшшшшш Й030НГЛГ 
63 52 0 


(—1)559n.2(€71023). (1 + |. 275) е е (0, 2047) 


1A HighLoad++ 
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ТЕЕЕ 754 


ехропеп! fraction 
sign (11 bit) (52 bit) 


I E SA 
А О 
63 52 0 


(—1)°#".2(е-—1023).(1 + |. 275?) е е (0, 2047) 
(1 | о(е-- 1023) | (0 На | | дз) ‚е=0 


1A HighLoad++ 
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ТЕЕЕ 764 


exponent fraction 
sign (11 bit) (52 bit) 


I 고 
ши 
63 52 0 


(—1)99.9(71029. (3 + f. 2-9?) е е (0,2047) 
= | о(е-- 1023) | (0 Us 1 | д) ‚е=0 


0 11111111111 00000000000000000000000000000000000000000000000000002 == +inf 


1 11111111111 00000000000000000000000000000000000000000000000000002 == -inf 
S 11111111111 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX2 == nan 


А HighLoad 
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Мога Number 


math.sgrt(-1) 


math.huge / math.huge 
0/0 


А HighLoad 
(ік анан 


Not a Number 


math.sqrt(-1) 
ath.huge / math.huge 


m 
0 


> nan -- false 
« nan -- false 
-- nan -- false 


-- true 


nan 
nan 
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Мога Number 


function assert де(1, г) 
if 1 <г then 


error("Assertion failed!") 
end 
end 


1^ HighLoad 


Not а Number 


function assert ge(l, г) 
if | < г then 
error("Assertion failed!") 
end 
end 


function assert ge(l, г) 

- if lL < г then 

+ if not (| >= г) then 
error("Assertion failed!") 


end 
end 
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Not a Number 


А HighLoad 
(ік анан 


Not а Number 


1 ^ пап -- 1 
пап ^ 0 -- 1 


тап ром 
SYNOPSIS 
#include <math.h> 
double pow(double x, double y); 


RETURN VALUE 


If x is +1, the result is 1.0 (even if y is a NaN). 
If y is 0, the result is 1.0 (even if x is a NaN). 


Linux man page 
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LuaJIT internal tags 


—. 
—. 


lj obj.h 


Interpreted as a double these are special NaNs. The FPU only generates 
one type of NaN (Oxfff8 0000 0000 0000). So MSWs » Oxfff80000 are available 
for use as internal tags. 


primitive types 
lightuserdata 

GC objects 

int (12 DUALNUM) 
number 


/ 
/ 
/ 
/ 
/ 
/ 
/ 
/ 
/ 


ЦМ 


#define LJ ТМП 
#define LJ TFALSE 
#define LJ TTRUE 
#define LJ TSTR 
#define 17 ТТАВ 


1^ HighLoad 
(нг) See 


Таблицы снаружи 


t1 = | 'Sunday', 'Monday', "Іп tired' | 
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Таблицы снаружи 


А HighLoad 
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Таблицы снаружи 
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Таблицы изнутри 


typedef struct GCtab { 
/* GC stuff */ 
MRef аггау; /« Аггау part. %/ 
MRef node; /* Hash part. */ 


uint32 t asize; /* Size of array part (keys |0, asize-1]). */ 
uint32 t hmask; /* Hash part mask (size of hash part - 1). */ 
) GCtab; 


г Highl оаа 
(нс) мана 


Таблицы изнутри 


typedef struct GCtab { 
/* GC stuff */ 


MRef node; /* Hash part. */ 


uint32 t hmask; /* Hash part mask (size of hash part - 1). */ 
) GCtab; 
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Таблицы изнутри 


typedef struct GCtab { 
/* GC stuff */ 
MRef array; /* Array part. */ 


uint32 t asize; /* Size of array part (keys |0, asize-1]). */ 


} GCtab; 


1^ HighLoad 
oa а 


Таблицы изнутри 


с = 4) 
- table: 0x40eae3a8 


a[0]: 
h[1]: nil=nil 


1^ HighLoad 
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Таблицы изнутри 


: 0x40eae3a8 


5111 


x40eae3a8 


* р-В, nilsnil, а-А, с-С 
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Таблицы изнутри 


(1 = {à = 1. Беа, € = 3) 
- table: 0x40eaeb08 
а[0]: 
h[4]: 0-2, nilsnil, asi, с=3 


t2 = {€ = 3, b = 2, а 
- table: 0x40ea7e70 
а[0]: 
h[4]: b=2, nil=nil, с=3, а=1 


1^ HighLoad 
ној atest 


Таблицы изнутри 


= ча - 7, с 
-- table: 0x40eaeb08 
а[0]: 
h[4]: 0-2, nilsnil, asi, с=3 


12 = {€ = 3, b = 2, à 
-- table: 0x40ea7e70 
а[0]: 
ћ[4]: b=2, nil=nil, c=3, а-1 


traverse(pairs, t1) 
-- b=2, a=1, c=3 


traverse(pairs, t2) 
-- b=2, c=3, a=1 
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Таблицы изнутри 


t2["c"] = nil 
- table: 0x411c83c0 


а[0]: 
h[4]: b=2, nil=nil, c=nil, azi 
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Таблицы изнутри 


Кај е" = nil 
- table: 0x411c83c0 


a[0]: 
11:11: bs2, nilsnil, c=nil, asi 


next(t2, "c") -- a 
next(t2, "d") -- error: invalid key to 'next' 
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Seguence 


t= 11, 2} 


table: 0x41735918 
a[3]* nil, 1, 2 
h[1]: nil=nil 


1^ HighLoad 
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Seguence 


tsi) 

-- table: 0x41735918 
a[3]: nil, 1, 2 
h[1]: nil=nil 


t = {[2] = 2, 1} 

-- table: 0x416a3998 
-- a[2]: nil, 1 

-- h[2]: nil=nil, 2-2 


Њ Highl саа 
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Seguence 


СЕТЕ, 2} 

-- table: 0x41735918 
8131: nil, 1, 2 
h[1]: nil=nil 


t = {[2] = 2, 1} 

-- table: 0x416a3998 
-- а[2]: nil, 1 

-- h[2]: nil=nil, 2-2 


t = table.new(4, 4) 

for i = 1, 8 do t[i] = i end 
-- table: 0x412c6df0 

-- a[5]: nil, 1, 2, 3, 4 

-- h[4]: 7=7, 8=8, 5=5, 6=6 
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Длина массива — определение 


Lua 51 Reference Manual. 85.4.6 — The Length Operator 


1A HighLoad+ 
(нь) нето 


Длина массива — определение 


#{1, 2, 3) -- 3 


Undefined behavior: 


- table: 0х41045528 
a[3]: nil, nil, 2 
h[1]: nil=nil 


- table: 0х41045810 
a[0]: 
h[2]: nilsntl, 222 


Lua 51 Reference Manual. 85.4.6 - The Length Operator 
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Откуда берутса дырки? 


„Ж = пъ 
+ table.remove(t, 1) 


1A HighLoad+ 
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Откуда берутся дырки? 


+ EI] = nk 
+ table.remove(t, 1) 


function vararg(...) 
local args = {...} 
- #args == undefined behavior 


end 


vararg(nil, "егг") 


1^ HighLoad 
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Длина массива — применение 
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Длина массива — применение 


table.sort(t) -- 1, #1 
unpack(t) -- 1, #t 


function table.pack(...) 
return (n = select('#', ...), ... 
end 


t = table.pack(nil, 2) 
unpack(t, 1, t.n) -- nil, 2 


1^ HighLoad 
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Итерации 


for 1 = 1, #t do 
епа 


for i, м in ipairs(t) do 
end 


1^ HighLoad 
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Итерации 


for 1 = 1, #t do 
епа 


for i, м in ipairs(t) do 
end 


-- pairs: 

local i = 1 

while type(t[i]) ~= 'nil' do 
-- do something 
1 =71+1 

епд 
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РЕЈ исааја 


ffi = require('ffi') 


NULL = ffi.new('void*', nil) 


А HighLoad 
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РЕЈ и cdata 


ffi = require('ffi') 
NULL = ffi.new('void*', nil) 


type(nil) -- nil 
type(NULL) -- cdata 
ffi.typeof(box.NULL) -- ctype<void *> 
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РЕЈ и cdata 


ffi = require('ffi') 
NULL = ffi.new('void*', nil) 


type(nil) -- nil 
type(NULL) -- cdata 
ffi.typeof(box.NULL) -- ctype<void *> 


NULL == nil -- true 


if NULL then 

print('NULL is not nil') 
end 
-- NULL is not nil 


Ч Highl саа 
oa 


РЕЈ исааја 


YA 


Выводы 


О 


Старайтесь писать код без багов © 
Проверяйте аргументы, пользуйтесь линтерами 
Избегайте Мам 


О 


О 


О 


Не делайте лишних предположений 


О 


Бойтесь дырявых массивов 
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(HL) de der 


